<?php

namespace PowerComponents\LivewirePowerGrid\DataSource\Support;

use Exception;
use Illuminate\Database\Eloquent\Builder as EloquentBuilder;
use Illuminate\Database\Query\Builder as QueryBuilder;
use Illuminate\Support\Facades\{DB, Schema};

class Sql
{
    /**
     * @var array<int,string>|string[]
     */
    private static array $sortStringNumberTypes = ['string', 'varchar', 'char'];

    /**
     * @throws Exception
     */
    public static function like(EloquentBuilder|QueryBuilder|null $query = null): string
    {
        if ($query) {
            /** @phpstan-ignore-next-line  */
            $driverName = $query->getConnection()->getConfig('driver');
        }

        if (! isset($driverName) or ! is_string($driverName)) {
            $driverName = self::getDatabaseDriverName();
        }

        /*
        |--------------------------------------------------------------------------
        |  Search 'LIKE' (drivers)
        |--------------------------------------------------------------------------
        | PowerGrid needs to sort with "case-insensitive".
        | Here, we adapt the 'LIKE' syntax for each database driver.
        */

        $likeSyntax = [
            'pgsql' => 'ILIKE',
        ];

        return $likeSyntax[$driverName] ?? 'LIKE';
    }

    /**
     * @throws Exception
     */
    public static function sortStringAsNumber(string $sortField): string
    {
        $driverName = self::getDatabaseDriverName();
        $driverVersion = self::getDatabaseVersion();

        return self::getSortSqlByDriver($sortField, $driverName, $driverVersion);
    }

    /**
     * @throws Exception
     */
    public static function getSortSqlByDriver(string $sortField, string $driverName = '', string $driverVersion = ''): string
    {
        if (empty($sortField) || empty($driverName) || empty($driverVersion)) {
            throw new Exception('sortField, driverName and driverVersion must be informed');
        }

        /*
        |--------------------------------------------------------------------------
        |  Supported Databases (drivers)
        |--------------------------------------------------------------------------
        | PowerGrid needs to sort string as number. I.e: Rooms  1, 1a, 1b, 2, 3... 10.
        | Here, we adapt the SQL sorting syntax between databases and versions.
        |  0     => default,
        |  x.x.x => version which the syntax was implemented.
        */

        $default = "$sortField+0 {sortDirection}"; // default, fallback

        $supportedVersions = [
            'mysql' => [
                '0' => $default,
                '8.0.4' => "CAST(NULLIF(REGEXP_REPLACE($sortField, '[[:alpha:]]+', ''), '') AS SIGNED INTEGER) {sortDirection}",
            ],
            'sqlite' => [
                '0' => "CAST($sortField AS INTEGER) {sortDirection}",
            ],
            'pgsql' => [
                '0' => "CAST(NULLIF(REGEXP_REPLACE($sortField, '\D', '', 'g'), '') AS INTEGER) {sortDirection}",
            ],
            'sqlsrv' => [
                '0' => "CAST(SUBSTRING($sortField, PATINDEX('%[a-z]%', $sortField), LEN($sortField)-PATINDEX('%[a-z]%', $sortField)) AS INT) {sortDirection}",
            ],
        ];

        if (! isset($supportedVersions[$driverName])) {
            return $default;
        }

        $syntax = collect($supportedVersions[$driverName])
            ->filter(function ($syntax, $version) use ($driverVersion) {
                /** @var string $version */
                return version_compare($version, $driverVersion, '<=');
            })
            ->last();

        return is_null($syntax) === true ? $default : $syntax;
    }

    public static function isValidSortFieldType(?string $sortFieldType): bool
    {
        if (is_null($sortFieldType)) {
            return false;
        }

        return in_array($sortFieldType, self::$sortStringNumberTypes);
    }

    /**
     * @throws Exception
     */
    public static function getSortFieldType(string $sortField): ?string
    {
        $data = explode('.', $sortField);

        if (! isset($data[1])) {
            return null;
        }

        $tableName = $data[0];
        $column = $data[1];

        if (! Schema::hasColumn($data[0], $data[1])) {
            throw new Exception("There is no column with name '$column' on table '$tableName'. Please see: https://livewire-powergrid.com/#/table/include-columns?id=fieldstring-field-string-datafield");
        }

        return Schema::getColumnType($tableName, $column);
    }

    /**
     * @throws Exception
     */
    public static function getDatabaseDriverName(): string
    {
        $driverName = DB::getDriverName();

        if (! is_string($driverName)) {
            throw new Exception('Could not get Database version');
        }

        return $driverName;
    }

    /**
     * @throws Exception
     */
    public static function getDatabaseVersion(): string
    {
        $version = DB::getPdo()->getAttribute(intval(constant('PDO::ATTR_SERVER_VERSION')));

        if (! is_string($version)) {
            throw new Exception('Could not get Database version');
        }

        return $version;
    }
}
